home *** CD-ROM | disk | FTP | other *** search
/ SPACE 2 / SPACE - Library 2 - Volume 1.iso / program / 515 / rcs5ap1s.lzh / RLOG.C < prev    next >
C/C++ Source or Header  |  1991-01-30  |  34KB  |  1,165 lines

  1. /*
  2.  *                       RLOG    operation
  3.  */
  4. /*****************************************************************************
  5.  *                       print contents of RCS files
  6.  *****************************************************************************
  7.  */
  8.  
  9. /* Copyright (C) 1982, 1988, 1989 Walter Tichy
  10.    Copyright 1990 by Paul Eggert
  11.    Distributed under license by the Free Software Foundation, Inc.
  12.  
  13. This file is part of RCS.
  14.  
  15. RCS is free software; you can redistribute it and/or modify
  16. it under the terms of the GNU General Public License as published by
  17. the Free Software Foundation; either version 1, or (at your option)
  18. any later version.
  19.  
  20. RCS is distributed in the hope that it will be useful,
  21. but WITHOUT ANY WARRANTY; without even the implied warranty of
  22. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  23. GNU General Public License for more details.
  24.  
  25. You should have received a copy of the GNU General Public License
  26. along with RCS; see the file COPYING.  If not, write to
  27. the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
  28.  
  29. Report problems and direct all questions to:
  30.  
  31.     rcs-bugs@cs.purdue.edu
  32.  
  33. */
  34.  
  35.  
  36.  
  37.  
  38. /* $Log: rlog.c,v $
  39.  * Revision 5.8  1991/01/30  14:21:32  apratt
  40.  * CI with RCS version 5
  41.  *
  42.  * Revision 5.7  91/01/30  12:02:48  apratt
  43.  * Changed RCS5AKP1 to RCS5AP1
  44.  * 
  45.  * Revision 5.6  91/01/29  17:45:26  apratt
  46.  * Added RCS5AKP1 to usage message
  47.  * 
  48.  * Revision 5.5  90/11/01  05:03:55  eggert
  49.  * checked in with -k by apratt at 91.01.10.13.15.34.
  50.  * 
  51.  * Revision 5.5  1990/11/01  05:03:55  eggert
  52.  * Permit arbitrary data in logs and comment leaders.
  53.  *
  54.  * Revision 5.4  1990/10/04  06:30:22  eggert
  55.  * Accumulate exit status across files.
  56.  *
  57.  * Revision 5.3  1990/09/11  02:41:16  eggert
  58.  * Plug memory leak.
  59.  *
  60.  * Revision 5.2  1990/09/04  08:02:33  eggert
  61.  * Count RCS lines better.
  62.  *
  63.  * Revision 5.0  1990/08/22  08:13:48  eggert
  64.  * Remove compile-time limits; use malloc instead.  Add setuid support.
  65.  * Switch to GMT.
  66.  * Report dates in long form, to warn about dates past 1999/12/31.
  67.  * Change "added/del" message to make room for the longer dates.
  68.  * Don't generate trailing white space.  Add -V.  Ansify and Posixate.
  69.  *
  70.  * Revision 4.7  89/05/01  15:13:48  narten
  71.  * changed copyright header to reflect current distribution rules
  72.  * 
  73.  * Revision 4.6  88/08/09  19:13:28  eggert
  74.  * Check for memory exhaustion; don't access freed storage.
  75.  * Shrink stdio code size; remove lint.
  76.  * 
  77.  * Revision 4.5  87/12/18  11:46:38  narten
  78.  * more lint cleanups (Guy Harris)
  79.  * 
  80.  * Revision 4.4  87/10/18  10:41:12  narten
  81.  * Updating version numbers
  82.  * Changes relative to 1.1 actually relative to 4.2
  83.  * 
  84.  * Revision 1.3  87/09/24  14:01:10  narten
  85.  * Sources now pass through lint (if you ignore printf/sprintf/fprintf 
  86.  * warnings)
  87.  * 
  88.  * Revision 1.2  87/03/27  14:22:45  jenkins
  89.  * Port to suns
  90.  * 
  91.  * Revision 4.2  83/12/05  09:18:09  wft
  92.  * changed rewriteflag to external.
  93.  * 
  94.  * Revision 4.1  83/05/11  16:16:55  wft
  95.  * Added -b, updated getnumericrev() accordingly.
  96.  * Replaced getpwuid() with getcaller().
  97.  * 
  98.  * Revision 3.7  83/05/11  14:24:13  wft
  99.  * Added options -L and -R;
  100.  * Fixed selection bug with -l on multiple files.
  101.  * Fixed error on dates of the form -d'>date' (rewrote getdatepair()).
  102.  * 
  103.  * Revision 3.6  82/12/24  15:57:53  wft
  104.  * shortened output format.
  105.  *
  106.  * Revision 3.5  82/12/08  21:45:26  wft
  107.  * removed call to checkaccesslist(); used DATEFORM to format all dates;
  108.  * removed unused variables.
  109.  *
  110.  * Revision 3.4  82/12/04  13:26:25  wft
  111.  * Replaced getdelta() with gettree(); removed updating of field lockedby.
  112.  *
  113.  * Revision 3.3  82/12/03  14:08:20  wft
  114.  * Replaced getlogin with getpwuid(), %02d with %.2d, fancydate with PRINTDATE.
  115.  * Fixed printing of nil, removed printing of Suffix,
  116.  * added shortcut if no revisions are printed, disambiguated struct members.
  117.  *
  118.  * Revision 3.2  82/10/18  21:09:06  wft
  119.  * call to curdir replaced with getfullRCSname(),
  120.  * fixed call to getlogin(), cosmetic changes on output,
  121.  * changed conflicting long identifiers.
  122.  *
  123.  * Revision 3.1  82/10/13  16:07:56  wft
  124.  * fixed type of variables receiving from getc() (char -> int).
  125.  */
  126.  
  127.  
  128.  
  129. #include "rcsbase.h"
  130.  
  131. struct  lockers {                     /* lockers in locker option; stored   */
  132.      const char         * login;      /* lockerlist                         */
  133.      struct     lockers * lockerlink;
  134.      }  ;
  135.  
  136. struct  stateattri {                  /* states in state option; stored in  */
  137.      const char         * status;     /* statelist                          */
  138.      struct  stateattri * nextstate;
  139.      }  ;
  140.  
  141. struct  authors {                     /* login names in author option;      */
  142.      const char         * login;      /* stored in authorlist               */
  143.      struct     authors * nextauthor;
  144.      }  ;
  145.  
  146. struct Revpairs{                      /* revision or branch range in -r     */
  147.      unsigned          numfld;     /* option; stored in revlist        */
  148.      const char         * strtrev;
  149.      const char         * endrev;
  150.      struct  Revpairs   * rnext;
  151.      } ;
  152.  
  153. struct Datepairs{                     /* date range in -d option; stored in */
  154.      char               strtdate[datesize];   /* duelst and datelist      */
  155.      char               enddate[datesize];
  156.      struct  Datepairs  * dnext;
  157.      };
  158.  
  159. static char extractdelta P((const struct hshentry*));
  160. static int checkrevpair P((const char*,const char*));
  161. static int readdeltalog P((void));
  162. static void cleanup P((void));
  163. static void extdate P((struct hshentry*));
  164. static void exttree P((struct hshentry*));
  165. static void getauthor P((char*));
  166. static void getdatepair P((char*));
  167. static void getlocker P((char*));
  168. static void getnumericrev P((void));
  169. static void getrevpairs P((char*));
  170. static void getscript P((struct hshentry*));
  171. static void getstate P((char*));
  172. static void putabranch P((const struct hshentry*));
  173. static void putadelta P((const struct hshentry*,const struct hshentry*,int));
  174. static void putforest P((const struct branchhead*));
  175. static void putree P((const struct hshentry*));
  176. static void putrunk P((void));
  177. static void recentdate P((const struct hshentry*,struct Datepairs*));
  178. static void trunclocks P((void));
  179.  
  180. static const char *insDelFormat;
  181. static int branchflag;    /*set on -b */
  182. static int exitstatus;
  183. static int lockflag;
  184. static int revno;    /* number of revision chosen */
  185. static struct Datepairs *datelist, *duelst;
  186. static struct Revpairs *revlist, *Revlst;
  187. static struct authors *authorlist;
  188. static struct lockers *lockerlist;
  189. static struct stateattri *statelist;
  190.  
  191.  
  192. mainProg(rlogId, "rlog", "$Id: rlog.c,v 5.8 1991/01/30 14:21:32 apratt Exp $")
  193. {
  194.     static const char cmdusage[] =
  195.         "\nRCS5AP1 as modified for TOS by Allan Pratt, atari!apratt\nrlog usage: rlog -{bhLRt} -ddates -l[lockers] -rrevs -sstates -w[logins] -Vn file ...";
  196.  
  197.     struct Datepairs *currdate;
  198.     const char *accessListString, *accessFormat, *commentFormat;
  199.     const char *headFormat, *symbolFormat;
  200.     const struct access *curaccess;
  201.     const struct assoc *curassoc;
  202.     const struct lock *currlock;
  203.     int descflag, selectflag;
  204.     int onlylockflag;  /* print only files with locks */
  205.     int selectop;  /* print only some revisions */
  206.     int onlyRCSflag;  /* print only RCS file name */
  207.  
  208.     initid();
  209.  
  210.         descflag = selectflag = true;
  211.     onlylockflag = selectop = onlyRCSflag = false;
  212.  
  213.         while (--argc,++argv, argc>=1 && ((*argv)[0] == '-')) {
  214.                 switch ((*argv)[1]) {
  215.  
  216.         case 'L':
  217.             onlylockflag = true;
  218.             break;
  219.  
  220.         case 'R':
  221.             onlyRCSflag =true;
  222.             break;
  223.  
  224.                 case 'l':
  225.                         selectop = true;
  226.                         lockflag = true;
  227.                         getlocker( (*argv)+2 );
  228.                         break;
  229.  
  230.                 case 'b':
  231.                         selectop = true;
  232.                         branchflag = true;
  233.                         break;
  234.  
  235.                 case 'r':
  236.                         selectop = true;
  237.                         getrevpairs( (*argv)+2 );
  238.                         break;
  239.  
  240.                 case 'd':
  241.                         selectop = true;
  242.                         getdatepair( (*argv)+2 );
  243.                         break;
  244.  
  245.                 case 's':
  246.                         selectop = true;
  247.                         getstate( (*argv)+2);
  248.                         break;
  249.  
  250.                 case 'w':
  251.                         selectop = true;
  252.                         getauthor( (*argv)+2);
  253.                         break;
  254.  
  255.                 case 'h':
  256.                         if ( ! selectflag ) warn("-t overrides -h.");
  257.                         else    descflag = false;
  258.                         break;
  259.  
  260.                 case 't':
  261.                         selectflag = false;
  262.                         if ( ! descflag ) warn("-t overrides -h.");
  263.                         descflag = true;
  264.                         break;
  265.  
  266.         case 'V':
  267.             setRCSversion(*argv);
  268.             break;
  269.  
  270.                 default:
  271.             faterror("unknown option: %s%s", *argv, cmdusage);
  272.  
  273.                 };
  274.         } /* end of option processing */
  275.  
  276.     if (argc<1) faterror("no input file%s", cmdusage);
  277.  
  278.     if (RCSversion < VERSION(5)) {
  279.         accessListString = "\naccess list:   ";
  280.         accessFormat = "  %s";
  281.         commentFormat = "\ncomment leader:  \"";
  282.         headFormat = "\nRCS file:        %s;   Working file:    %s\nhead:           %s%s\nbranch:         %s%s\nlocks:         ";
  283.         insDelFormat = "  lines added/del: %lu/%lu";
  284.         symbolFormat = "  %s: %s;";
  285.     } else {
  286.         accessListString = "\naccess list:";
  287.         accessFormat = "\n\t%s";
  288.         commentFormat = "\ncomment leader: \"";
  289.         headFormat = "\nRCS file: %s\nWorking file: %s\nhead:%s%s\nbranch:%s%s\nlocks:%s";
  290.         insDelFormat = "  lines: +%lu -%lu";
  291.         symbolFormat = "\n\t%s: %s";
  292.     }
  293.  
  294.         /* now handle all filenames */
  295.         do {
  296.         finptr = NULL;
  297.         ffree();
  298.  
  299.         if (!pairfilenames(argc, argv, rcsreadopen, true, false))
  300.         continue;
  301.  
  302.             /* now RCSfilename contains the name of the RCS file, and finptr
  303.              * the file descriptor. Workfilename contains the name of the
  304.              * working file.
  305.              */
  306.  
  307.         /* Keep only those locks given by -l.  */
  308.         if (lockflag)
  309.         trunclocks();
  310.  
  311.             /* do nothing if -L is given and there are no locks*/
  312.         if (onlylockflag && !Locks)
  313.         continue;
  314.  
  315.         if ( onlyRCSflag ) {
  316.         aprintf(stdout, "%s\n", RCSfilename);
  317.         continue;
  318.         }
  319.             /*   print RCS filename , working filename and optional
  320.                  administrative information                         */
  321.             /* could use getfullRCSname() here, but that is very slow */
  322.         aprintf(stdout, headFormat, RCSfilename, workfilename,
  323.             Head ? " " : "",  Head ? Head->num : "",
  324.             Dbranch ? " " : "",  Dbranch ? Dbranch : "",
  325.             StrictLocks ? " strict" : ""
  326.         );
  327.             currlock = Locks;
  328.             while( currlock ) {
  329.         aprintf(stdout, symbolFormat, currlock->login,
  330.                                 currlock->delta->num);
  331.                 currlock = currlock->nextlock;
  332.             }
  333.             if (StrictLocks && RCSversion<VERSION(5))
  334.         aputs("  strict", stdout);
  335.  
  336.         aputs(accessListString, stdout);      /*  print access list  */
  337.             curaccess = AccessList;
  338.             while(curaccess) {
  339.         aprintf(stdout, accessFormat, curaccess->login);
  340.                 curaccess = curaccess->nextaccess;
  341.             }
  342.  
  343.         aputs("\nsymbolic names:", stdout);   /*  print symbolic names   */
  344.         for (curassoc=Symbols; curassoc; curassoc=curassoc->nextassoc)
  345.         aprintf(stdout, symbolFormat, curassoc->symbol, curassoc->num);
  346.         aputs(commentFormat, stdout);
  347.         awrite(Comment.string, Comment.size, stdout);
  348.         aputs("\"\n", stdout);
  349.         if (VERSION(5)<=RCSversion  ||  Expand != KEYVAL_EXPAND)
  350.         aprintf(stdout, "keyword substitution: %s\n",
  351.             expand_names[Expand]
  352.         );
  353.  
  354.             gettree();
  355.  
  356.         aprintf(stdout, "total revisions: %d", TotalDeltas);
  357.  
  358.             if ( Head == nil || !selectflag || !descflag) {
  359.         afputc('\n',stdout);
  360.         if (descflag) aputs("description:\n", stdout);
  361.                 getdesc(descflag);
  362.         goto rlogend;
  363.             }
  364.  
  365.  
  366.             getnumericrev();    /* get numeric revision or branch names */
  367.             revno = 0;
  368.  
  369.             exttree(Head);
  370.  
  371.             /*  get most recently date of the dates pointed by duelst  */
  372.             currdate = duelst;
  373.             while( currdate) {
  374.                 recentdate(Head, currdate);
  375.                 currdate = currdate->dnext;
  376.         }
  377.  
  378.             extdate(Head);
  379.  
  380.             /*  reinitialize the date specification list   */
  381.             currdate = duelst;
  382.             while(currdate) {
  383.                 VOID sprintf(currdate->strtdate,DATEFORM,0,0,0,0,0,0);
  384.                 currdate = currdate->dnext;
  385.             }
  386.  
  387.             if ( selectop || ( selectflag && descflag) )
  388.         aprintf(stdout, ";\tselected revisions: %d", revno);
  389.         afputc('\n', stdout);
  390.         if (descflag) aputs("description:\n", stdout);
  391.             getdesc(descflag);
  392.             if (selectflag && descflag && revno) {
  393.         while (readdeltalog())
  394.             ;
  395.                 putrunk();
  396.                 putree(Head);
  397.         if (nexttok != EOFILE)
  398.             fatserror("expecting EOF");
  399.             }
  400.     rlogend:
  401.         aputs("=============================================================================\n",stdout);
  402.     } while (cleanup(),
  403.          ++argv, --argc >= 1);
  404.     exitmain(exitstatus);
  405. }
  406.  
  407.     static void
  408. cleanup()
  409. {
  410.     if (nerror) exitstatus = EXIT_FAILURE;
  411.     if (finptr) ffclose(finptr);
  412. }
  413.  
  414. #if lint
  415. #    define exiterr rlogExit
  416. #endif
  417.     exiting void
  418. exiterr()
  419. {
  420.     _exit(EXIT_FAILURE);
  421. }
  422.  
  423.  
  424.  
  425.     static void
  426. putrunk()
  427. /*  function:  print revisions chosen, which are in trunk      */
  428.  
  429. {
  430.     register const struct hshentry *ptr;
  431.  
  432.     for (ptr = Head;  ptr;  ptr = ptr->next)
  433.         putadelta(ptr, ptr->next, true);
  434. }
  435.  
  436.  
  437.  
  438.     static void
  439. putree(root)
  440.     const struct hshentry *root;
  441. /*   function: print delta tree (not including trunk) in reverse
  442.                order on each branch                                        */
  443.  
  444. {
  445.         if ( root == nil ) return;
  446.  
  447.         putree(root->next);
  448.  
  449.         putforest(root->branches);
  450. }
  451.  
  452.  
  453.  
  454.  
  455.     static void
  456. putforest(branchroot)
  457.     const struct branchhead *branchroot;
  458. /*   function:  print branches that has the same direct ancestor    */
  459. {
  460.  
  461.         if ( branchroot == nil ) return;
  462.  
  463.         putforest(branchroot->nextbranch);
  464.  
  465.         putabranch(branchroot->hsh);
  466.         putree(branchroot->hsh);
  467. }
  468.  
  469.  
  470.  
  471.  
  472.     static void
  473. putabranch(root)
  474.     const struct hshentry *root;
  475. /*   function  :  print one branch     */
  476.  
  477. {
  478.  
  479.         if ( root == nil) return;
  480.  
  481.         putabranch(root->next);
  482.  
  483.         putadelta(root, root, false);
  484. }
  485.  
  486.  
  487.  
  488.  
  489.  
  490.     static void
  491. putadelta(node,editscript,trunk)
  492.     register const struct hshentry *node, *editscript;
  493.     int trunk;
  494. /*  function: Print delta node if node->selector is set.        */
  495. /*      editscript indicates where the editscript is stored     */
  496. /*      trunk indicated whether this node is in trunk           */
  497. {
  498.     const struct branchhead *newbranch;
  499.     struct buf branchnum;
  500.  
  501.     if (!node->selector)
  502.             return;
  503.  
  504.     aprintf(stdout,
  505.         "----------------------------\nrevision %s", node->num
  506.     );
  507.         if ( node->lockedby )
  508.        aprintf(stdout, "\tlocked by: %s;", node->lockedby);
  509.  
  510.     aputs("\ndate: ",stdout);
  511.     printdate(stdout, node->date, " ");
  512.     aprintf(stdout, ";  author: %s;  state: %s;",
  513.         node->author, node->state
  514.     );
  515.  
  516.         if ( editscript )
  517.            if(trunk)
  518.           aprintf(stdout, insDelFormat,
  519.                              editscript->deletelns, editscript->insertlns);
  520.            else
  521.           aprintf(stdout, insDelFormat,
  522.                              editscript->insertlns, editscript->deletelns);
  523.  
  524.         newbranch = node->branches;
  525.         if ( newbranch ) {
  526.        bufautobegin(&branchnum);
  527.        aputs("\nbranches:", stdout);
  528.            while( newbranch ) {
  529.         getbranchno(newbranch->hsh->num, &branchnum);
  530.         aprintf(stdout, "  %s;", branchnum.string);
  531.                 newbranch = newbranch->nextbranch;
  532.            }
  533.        bufautoend(&branchnum);
  534.         }
  535.  
  536.     afputc('\n', stdout);
  537.     awrite(node->log.string, node->log.size, stdout);
  538. }
  539.  
  540.  
  541.  
  542.  
  543.  
  544.     static int
  545. readdeltalog()
  546. /*  Function : get the log message and skip the text of a deltatext node.
  547.  *             Return false if current block does not start with a number.
  548.  *             Assumes the current lexeme is not yet in nexttok; does not
  549.  *             advance nexttok.
  550.  */
  551. {
  552.         register struct  hshentry  * Delta;
  553.     struct buf logbuf;
  554.  
  555.         nextlex();
  556.         if ( !(Delta = getnum() )) return(false);
  557.     getkeystring(Klog);
  558.     bufautobegin(&logbuf);
  559.     Delta->log = savestring(&logbuf);
  560.     /*
  561.      * Do the following instead of bufautoend(&logbuf),
  562.      * because the buffer must survive until we are done with the file.
  563.      */
  564.     Delta->log.string = (char *)fremember(testrealloc(
  565.         (malloc_type)logbuf.string,
  566.         Delta->log.size
  567.     ));
  568.  
  569.         nextlex();
  570.     while (nexttok==ID && strcmp(NextString,Ktext)!=0)
  571.         ignorephrase();
  572.     getkeystring(Ktext);
  573.         Delta->insertlns = Delta->deletelns = 0;
  574.         if ( Delta != Head)
  575.                 getscript(Delta);
  576.         else
  577.                 readstring();
  578.         return true;
  579. }
  580.  
  581.  
  582.  
  583.     static void
  584. getscript(Delta)
  585. struct    hshentry   * Delta;
  586. /*   function:  read edit script of Delta and count how many lines added  */
  587. /*              and deleted in the script                                 */
  588.  
  589. {
  590.         int ed;   /*  editor command  */
  591.     register FILE * fin;
  592.         register  int   c;
  593.     register unsigned long i;
  594.     struct diffcmd dc;
  595.  
  596.     fin = finptr;
  597.     initdiffcmd(&dc);
  598.     while (0  <=  (ed = getdiffcmd(fin,SDELIM,(FILE *)0,&dc)))
  599.         if (!ed)
  600.                  Delta->deletelns += dc.nlines;
  601.         else {
  602.                  /*  skip scripted lines  */
  603.          i = dc.nlines;
  604.          Delta->insertlns += i;
  605.          do {
  606.              while ((c=getc(fin)) != '\n')
  607.             if (c==EOF  ||  c==SDELIM && (c=getc(fin))!=SDELIM) {
  608.                 if (c==EOF || i!=1)
  609.                 fatserror("unexpected end to edit script");
  610.                 nextc = c;
  611.                 return;
  612.             }
  613.              ++rcsline;
  614.          } while (--i);
  615.             }
  616.     nextc = getc(fin);
  617. }
  618.  
  619.  
  620.  
  621.  
  622.  
  623.  
  624.  
  625.     static void
  626. exttree(root)
  627. struct hshentry  *root;
  628. /*  function: select revisions , starting with root             */
  629.  
  630. {
  631.     const struct branchhead *newbranch;
  632.  
  633.         if (root == nil) return;
  634.  
  635.     root->selector = extractdelta(root);
  636.         exttree(root->next);
  637.  
  638.         newbranch = root->branches;
  639.         while( newbranch ) {
  640.             exttree(newbranch->hsh);
  641.             newbranch = newbranch->nextbranch;
  642.         }
  643. }
  644.  
  645.  
  646.  
  647.  
  648.     static void
  649. getlocker(argv)
  650. char    * argv;
  651. /*   function : get the login names of lockers from command line   */
  652. /*              and store in lockerlist.                           */
  653.  
  654. {
  655.         register char c;
  656.         struct   lockers   * newlocker;
  657.         argv--;
  658.         while( ( c = (*++argv)) == ',' || c == ' ' || c == '\t' ||
  659.                  c == '\n' || c == ';')  ;
  660.         if (  c == '\0') {
  661.             lockerlist=nil;
  662.             return;
  663.         }
  664.  
  665.         while( c != '\0' ) {
  666.         newlocker = talloc(struct lockers);
  667.             newlocker->lockerlink = lockerlist;
  668.             newlocker->login = argv;
  669.             lockerlist = newlocker;
  670.             while ( ( c = (*++argv)) != ',' && c != '\0' && c != ' '
  671.                        && c != '\t' && c != '\n' && c != ';') ;
  672.             *argv = '\0';
  673.             if ( c == '\0' ) return;
  674.             while( ( c = (*++argv)) == ',' || c == ' ' || c == '\t' ||
  675.                      c == '\n' || c == ';')  ;
  676.         }
  677. }
  678.  
  679.  
  680.  
  681.     static void
  682. getauthor(argv)
  683. char   *argv;
  684. /*   function:  get the author's name from command line   */
  685. /*              and store in authorlist                   */
  686.  
  687. {
  688.         register    c;
  689.         struct     authors  * newauthor;
  690.  
  691.         argv--;
  692.         while( ( c = (*++argv)) == ',' || c == ' ' || c == '\t' ||
  693.                  c == '\n' || c == ';')  ;
  694.         if ( c == '\0' ) {
  695.         authorlist = talloc(struct authors);
  696.         authorlist->login = getcaller();
  697.             authorlist->nextauthor  = nil;
  698.             return;
  699.         }
  700.  
  701.         while( c != '\0' ) {
  702.         newauthor = talloc(struct authors);
  703.             newauthor->nextauthor = authorlist;
  704.             newauthor->login = argv;
  705.             authorlist = newauthor;
  706.             while( ( c = *++argv) != ',' && c != '\0' && c != ' '
  707.                      && c != '\t' && c != '\n' && c != ';') ;
  708.             * argv = '\0';
  709.             if ( c == '\0') return;
  710.             while( ( c = (*++argv)) == ',' || c == ' ' || c == '\t' ||
  711.                      c == '\n' || c == ';')  ;
  712.         }
  713. }
  714.  
  715.  
  716.  
  717.  
  718.     static void
  719. getstate(argv)
  720. char   * argv;
  721. /*   function :  get the states of revisions from command line  */
  722. /*               and store in statelist                         */
  723.  
  724. {
  725.         register  char  c;
  726.         struct    stateattri    *newstate;
  727.  
  728.         argv--;
  729.         while( ( c = (*++argv)) == ',' || c == ' ' || c == '\t' ||
  730.                  c == '\n' || c == ';')  ;
  731.         if ( c == '\0'){
  732.         warn("missing state attributes after -s options");
  733.             return;
  734.         }
  735.  
  736.         while( c != '\0' ) {
  737.         newstate = talloc(struct stateattri);
  738.             newstate->nextstate = statelist;
  739.             newstate->status = argv;
  740.             statelist = newstate;
  741.             while( (c = (*++argv)) != ',' && c != '\0' && c != ' '
  742.                     && c != '\t' && c != '\n' && c != ';')  ;
  743.             *argv = '\0';
  744.             if ( c == '\0' ) return;
  745.             while( ( c = (*++argv)) == ',' || c == ' ' || c == '\t' ||
  746.                      c == '\n' || c == ';')  ;
  747.         }
  748. }
  749.  
  750.  
  751.  
  752.     static void
  753. trunclocks()
  754. /*  Function:  Truncate the list of locks to those that are held by the  */
  755. /*             id's on lockerlist. Do not truncate if lockerlist empty.  */
  756.  
  757. {
  758.     const struct lockers *plocker;
  759.         struct lock     * plocked,  * nextlocked;
  760.  
  761.         if ( (lockerlist == nil) || (Locks == nil)) return;
  762.  
  763.         /* shorten Locks to those contained in lockerlist */
  764.         plocked = Locks;
  765.         Locks = nil;
  766.         while( plocked != nil) {
  767.             plocker = lockerlist;
  768.             while((plocker != nil) && ( strcmp(plocker->login, plocked->login)!=0))
  769.                 plocker = plocker->lockerlink;
  770.             nextlocked = plocked->nextlock;
  771.             if ( plocker != nil) {
  772.                 plocked->nextlock = Locks;
  773.                 Locks = plocked;
  774.             }
  775.             plocked = nextlocked;
  776.         }
  777. }
  778.  
  779.  
  780.  
  781.     static void
  782. recentdate(root, pd)
  783.     const struct hshentry *root;
  784.     struct Datepairs *pd;
  785. /*  function:  Finds the delta that is closest to the cutoff date given by   */
  786. /*             pd among the revisions selected by exttree.                   */
  787. /*             Successively narrows down the interval given by pd,           */
  788. /*             and sets the strtdate of pd to the date of the selected delta */
  789. {
  790.     const struct branchhead *newbranch;
  791.  
  792.     if ( root == nil) return;
  793.     if (root->selector) {
  794.              if ( cmpnum(root->date, pd->strtdate) >= 0 &&
  795.                   cmpnum(root->date, pd->enddate) <= 0)
  796.         VOID strcpy(pd->strtdate, root->date);
  797.         }
  798.  
  799.         recentdate(root->next, pd);
  800.         newbranch = root->branches;
  801.         while( newbranch) {
  802.            recentdate(newbranch->hsh, pd);
  803.            newbranch = newbranch->nextbranch;
  804.     }
  805. }
  806.  
  807.  
  808.  
  809.  
  810.  
  811.  
  812.     static void
  813. extdate(root)
  814. struct  hshentry        * root;
  815. /*  function:  select revisions which are in the date range specified     */
  816. /*             in duelst  and datelist, start at root                     */
  817.  
  818. {
  819.     const struct branchhead *newbranch;
  820.     const struct Datepairs *pdate;
  821.  
  822.         if ( root == nil) return;
  823.  
  824.         if ( datelist || duelst) {
  825.             pdate = datelist;
  826.             while( pdate ) {
  827.                 if ( (pdate->strtdate)[0] == '\0' || cmpnum(root->date,pdate->strtdate) >= 0){
  828.                    if ((pdate->enddate)[0] == '\0' || cmpnum(pdate->enddate,root->date) >= 0)
  829.                         break;
  830.                 }
  831.                 pdate = pdate->dnext;
  832.             }
  833.             if ( pdate == nil) {
  834.                 pdate = duelst;
  835.         for (;;) {
  836.            if (!pdate) {
  837.             root->selector = false;
  838.             break;
  839.            }
  840.                    if ( cmpnum(root->date, pdate->strtdate) == 0)
  841.                       break;
  842.                    pdate = pdate->dnext;
  843.                 }
  844.             }
  845.         }
  846.     if (root->selector)
  847.         ++revno;
  848.  
  849.         extdate(root->next);
  850.  
  851.         newbranch = root->branches;
  852.         while( newbranch ) {
  853.            extdate(newbranch->hsh);
  854.            newbranch = newbranch->nextbranch;
  855.         }
  856. }
  857.  
  858.  
  859.  
  860.     static char
  861. extractdelta(pdelta)
  862.     const struct hshentry *pdelta;
  863. /*  function:  compare information of pdelta to the authorlist, lockerlist,*/
  864. /*             statelist, revlist and yield true if pdelta is selected.    */
  865.  
  866. {
  867.     const struct lock *plock;
  868.     const struct stateattri *pstate;
  869.     const struct authors *pauthor;
  870.     const struct Revpairs *prevision;
  871.     unsigned length;
  872.  
  873.     if ((pauthor = authorlist)) /* only certain authors wanted */
  874.         while (strcmp(pauthor->login, pdelta->author) != 0)
  875.         if (!(pauthor = pauthor->nextauthor))
  876.             return false;
  877.     if ((pstate = statelist)) /* only certain states wanted */
  878.         while (strcmp(pstate->status, pdelta->state) != 0)
  879.         if (!(pstate = pstate->nextstate))
  880.             return false;
  881.     if (lockflag) /* only locked revisions wanted */
  882.         for (plock = Locks;  ;  plock = plock->nextlock)
  883.         if (!plock)
  884.             return false;
  885.         else if (plock->delta == pdelta)
  886.             break;
  887.     if ((prevision = Revlst)) /* only certain revs or branches wanted */
  888.         for (;;) {
  889.                 length = prevision->numfld;
  890.         if (
  891.             countnumflds(pdelta->num) == length+(length&1) &&
  892.             0 <= compartial(pdelta->num, prevision->strtrev, length) &&
  893.             0 <= compartial(prevision->endrev, pdelta->num, length)
  894.         )
  895.              break;
  896.         if (!(prevision = prevision->rnext))
  897.             return false;
  898.             }
  899.     return true;
  900. }
  901.  
  902.  
  903.  
  904.     static void
  905. getdatepair(argv)
  906.    char   * argv;
  907. /*  function:  get time range from command line and store in datelist if    */
  908. /*             a time range specified or in duelst if a time spot specified */
  909.  
  910. {
  911.         register   char         c;
  912.         struct     Datepairs    * nextdate;
  913.     const char        * rawdate;
  914.     int                     switchflag;
  915.  
  916.         argv--;
  917.         while( ( c = (*++argv)) == ',' || c == ' ' || c == '\t' ||
  918.                  c == '\n' || c == ';')  ;
  919.         if ( c == '\0' ) {
  920.         warn("missing date/time after -d");
  921.             return;
  922.         }
  923.  
  924.         while( c != '\0' )  {
  925.         switchflag = false;
  926.         nextdate = talloc(struct Datepairs);
  927.             if ( c == '<' ) {   /*   case: -d <date   */
  928.                 c = *++argv;
  929.                 (nextdate->strtdate)[0] = '\0';
  930.         } else if (c == '>') { /* case: -d'>date' */
  931.         c = *++argv;
  932.         (nextdate->enddate)[0] = '\0';
  933.         switchflag = true;
  934.         } else {
  935.                 rawdate = argv;
  936.         while( c != '<' && c != '>' && c != ';' && c != '\0')
  937.              c = *++argv;
  938.                 *argv = '\0';
  939.         if ( c == '>' ) switchflag=true;
  940.         str2date(rawdate,
  941.              switchflag ? nextdate->enddate : nextdate->strtdate);
  942.         if ( c == ';' || c == '\0') {  /*  case: -d date  */
  943.             VOID strcpy(nextdate->enddate,nextdate->strtdate);
  944.             VOID sprintf(nextdate->strtdate,DATEFORM,0,0,0,0,0,0);
  945.                     nextdate->dnext = duelst;
  946.                     duelst = nextdate;
  947.             goto end;
  948.         } else {
  949.             /*   case:   -d date<  or -d  date>; see switchflag */
  950.             while ( (c= *++argv) == ' ' || c=='\t' || c=='\n');
  951.             if ( c == ';' || c == '\0') {
  952.             /* second date missing */
  953.             if (switchflag)
  954.                 *nextdate->strtdate= '\0';
  955.             else
  956.                 *nextdate->enddate= '\0';
  957.             nextdate->dnext = datelist;
  958.             datelist = nextdate;
  959.             goto end;
  960.             }
  961.                 }
  962.             }
  963.             rawdate = argv;
  964.         while( c != '>' && c != '<' && c != ';' && c != '\0')
  965.          c = *++argv;
  966.             *argv = '\0';
  967.         str2date(rawdate,
  968.              switchflag ? nextdate->strtdate : nextdate->enddate);
  969.             nextdate->dnext = datelist;
  970.         datelist = nextdate;
  971.      end:
  972.         if ( c == '\0')  return;
  973.             while( (c = *++argv) == ';' || c == ' ' || c == '\t' || c =='\n');
  974.         }
  975. }
  976.  
  977.  
  978.  
  979.     static void
  980. getnumericrev()
  981. /*  function:  get the numeric name of revisions which stored in revlist  */
  982. /*             and then stored the numeric names in Revlst                */
  983. /*             if branchflag, also add default branch                     */
  984.  
  985. {
  986.         struct  Revpairs        * ptr, *pt;
  987.     unsigned n;
  988.     struct buf s, e;
  989.     const struct buf *rstart, *rend;
  990.  
  991.         Revlst = nil;
  992.         ptr = revlist;
  993.     bufautobegin(&s);
  994.     bufautobegin(&e);
  995.         while( ptr ) {
  996.         n = 0;
  997.         rstart = &s;
  998.         rend = &e;
  999.  
  1000.         switch (ptr->numfld) {
  1001.  
  1002.           case 1: /* -r rev */
  1003.         if (expandsym(ptr->strtrev, &s)) {
  1004.             rend = &s;
  1005.             n = countnumflds(s.string);
  1006.                 }
  1007.         break;
  1008.  
  1009.           case 2: /* -r rev- */
  1010.         if (expandsym(ptr->strtrev, &s)) {
  1011.             bufscpy(&e, s.string);
  1012.             n = countnumflds(s.string);
  1013.             (n<2 ? e.string : strrchr(e.string,'.'))[0]  =  0;
  1014.                 }
  1015.         break;
  1016.  
  1017.           case 3: /* -r -rev */
  1018.         if (expandsym(ptr->endrev, &e)) {
  1019.             if ((n = countnumflds(e.string)) < 2)
  1020.             bufscpy(&s, ".1");
  1021.             else {
  1022.             bufscpy(&s, e.string);
  1023.             VOID strcpy(strrchr(s.string,'.'), ".1");
  1024.             }
  1025.                 }
  1026.         break;
  1027.  
  1028.           default: /* -r rev1-rev2 */
  1029.         if (
  1030.             expandsym(ptr->strtrev, &s)
  1031.             &&    expandsym(ptr->endrev, &e)
  1032.             &&    checkrevpair(s.string, e.string)
  1033.         ) {
  1034.             n = countnumflds(s.string);
  1035.             /* Swap if out of order.  */
  1036.             if (compartial(s.string,e.string,n) > 0) {
  1037.             rstart = &e;
  1038.             rend = &s;
  1039.             }
  1040.         }
  1041.         break;
  1042.         }
  1043.  
  1044.         if (n) {
  1045.         pt = ftalloc(struct Revpairs);
  1046.         pt->numfld = n;
  1047.         pt->strtrev = fstrsave(rstart->string);
  1048.         pt->endrev = fstrsave(rend->string);
  1049.                 pt->rnext = Revlst;
  1050.                 Revlst = pt;
  1051.         }
  1052.         ptr = ptr->rnext;
  1053.         }
  1054.         /* Now take care of branchflag */
  1055.     if (branchflag && (Dbranch||Head)) {
  1056.         pt = ftalloc(struct Revpairs);
  1057.         pt->strtrev = pt->endrev =
  1058.         Dbranch ? Dbranch : fstrsave(partialno(&s,Head->num,1));
  1059.         pt->rnext=Revlst; Revlst=pt;
  1060.         pt->numfld = countnumflds(pt->strtrev);
  1061.         }
  1062.     bufautoend(&s);
  1063.     bufautoend(&e);
  1064. }
  1065.  
  1066.  
  1067.  
  1068.     static int
  1069. checkrevpair(num1,num2)
  1070.     const char *num1, *num2;
  1071. /*  function:  check whether num1, num2 are legal pair,i.e.
  1072.     only the last field are different and have same number of
  1073.     fields( if length <= 2, may be different if first field)   */
  1074.  
  1075. {
  1076.     unsigned length = countnumflds(num1);
  1077.  
  1078.     if (
  1079.             countnumflds(num2) != length
  1080.         ||    2 < length  &&  compartial(num1, num2, length-1) != 0
  1081.     ) {
  1082.         error("invalid branch or revision pair %s : %s", num1, num2);
  1083.             return false;
  1084.         }
  1085.  
  1086.         return true;
  1087. }
  1088.  
  1089.  
  1090.  
  1091.     static void
  1092. getrevpairs(argv)
  1093. register     char    * argv;
  1094. /*  function:  get revision or branch range from command line, and   */
  1095. /*             store in revlist                                      */
  1096.  
  1097. {
  1098.         register    char    c;
  1099.         struct      Revpairs  * nextrevpair;
  1100.         int         flag;
  1101.  
  1102.         argv--;
  1103.         while( ( c = (*++argv)) == ',' || c == ' ' || c == '\t' ||
  1104.                  c == '\n' || c == ';')  ;
  1105.         if ( c == '\0' ) {
  1106.         warn("missing revision or branch number after -r");
  1107.             return;
  1108.         }
  1109.  
  1110.         while( c != '\0') {
  1111.             while(  c  == ',' || c == ' ' || c == '\t' ||
  1112.                      c == '\n' || c == ';') c = *++argv;
  1113.             if (c == '\0')  return;
  1114.         nextrevpair = talloc(struct Revpairs);
  1115.             nextrevpair->rnext = revlist;
  1116.             revlist = nextrevpair;
  1117.         nextrevpair->numfld = 0;
  1118.             nextrevpair->strtrev = nil;
  1119.             nextrevpair->endrev  = nil;
  1120.             flag = false;
  1121.             if (  c == '<' || c == '-' ) {  /*  case: -r -rev  or -r <rev  */
  1122.                 flag = true;
  1123.                 while( (c =(*++argv)) == ' ' || c == '\t' || c =='\n') ;
  1124.             }
  1125.             else {
  1126.                 nextrevpair->strtrev = argv;
  1127.                 /*   get a revision or branch name  */
  1128.                 while( c != ',' && c != ';' && c != ' ' && c != '\0' && c != '-'
  1129.                         && c != '\t' && c != '\n' && c != '<') c = *++argv;
  1130.  
  1131.                 *argv = '\0';
  1132.  
  1133.                 if ( c != '<' && c != '-') {    /*  case: rev  */
  1134.                     nextrevpair->numfld = 1;
  1135.                     continue;
  1136.                 }
  1137.  
  1138.                 if ( (c =(*++argv)) == ',' || c == '\0' || c == ' '
  1139.                       || c == '\t' || c == '\n' || c == ';') {/*  case: rev_  */
  1140.                     nextrevpair->numfld = 2;
  1141.                     continue;
  1142.                 }
  1143.             }
  1144.             nextrevpair->endrev = argv;
  1145.             while( c != ',' && c != ' ' && c != '\0' && c != '\t' && c != '<'
  1146.                    && c != '\n' && c != '-' && c != ';')  c = *++argv;
  1147.  
  1148.             * argv = '\0';
  1149.             if ( c == '<'){
  1150.         error("separator expected near %s", nextrevpair->endrev);
  1151.                 while( (c = *++argv) != ',' && c != ' ' && c != '\0' &&
  1152.                         c != '\t' && c != '\n' && c != ';' ) ;
  1153.                 revlist = nextrevpair->rnext;
  1154.                 continue;
  1155.             }
  1156.             else  {
  1157.                 if (flag)   /*  case:  -rev   */
  1158.                     nextrevpair->numfld  = 3;
  1159.  
  1160.                 else     /*   rev1-rev2  appears  */
  1161.                     nextrevpair->numfld = 4;
  1162.             }
  1163.         }
  1164. }
  1165.